SpringBoot整合Activiti工作流(附源码) | 您所在的位置:网站首页 › Springboot 整合activiti及使用案例 › SpringBoot整合Activiti工作流(附源码) |
点击上方 "程序员小乐"关注, 星标或置顶一起成长 每天凌晨00点00分, 第一时间与你相约 每日英文 Never expect. Never assume. And never demand. Just let it be. If it's meant to be, it will happen. 不要期待,不要假想,不要强求,顺其自然,如果注定,便一定会发生。 每日掏心话 我们所回首的,都是在回望的凝视里静默。红尘的故事里,我是怎样的执着,执着到守望的等待永恒。 来自:yawn-silence | 责编:乐乐 链接:my.oschina.net/silenceyawen/blog/1609603 程序员小乐(ID:study_tech)第 729 次推文 图片来自网络 往日回顾:重磅!12306又双叒崩了!12306是不是世界上最复杂的系统? 正文 依赖:新建springBoot项目时勾选activiti,或者在已建立的springBoot项目添加以下依赖: org.activiti activiti-spring-boot-starter-basic 6.0.0 配置:数据源和activiti配置: server: port: 8081 spring: datasource: url: jdbc:mysql://localhost:3306/act5?useSSL=true driver-class-name: com.mysql.jdbc.Driver username: root password: root # activiti default configuration activiti: database-schema-update: true check-process-definitions: true process-definition-location-prefix: classpath:/processes/ # process-definition-location-suffixes: # - **.bpmn # - **.bpmn20.xml history-level: full在activiti的默认配置中,process-definition-location-prefix 是指定activiti流程描述文件的前缀(即路径),启动时,activiti就会去寻找此路径下的流程描述文件,并且自动部署;suffix 是一个String数组,表示描述文件的默认后缀名,默认以上两种。 springMVC配置: package com.yawn.config; import org.springframework.context.annotation.Configuration; import org.springframework.format.FormatterRegistry; import org.springframework.http.HttpStatus; import org.springframework.web.servlet.config.annotation.*; /** * Created by yawn on 2017/8/5. */ @EnableWebMvc @Configuration public class MvcConfig extends WebMvcConfigurerAdapter { @Override public void addResourceHandlers(ResourceHandlerRegistry registry) { registry.addResourceHandler("/static/**").addResourceLocations("classpath:/static/"); registry.addResourceHandler("/templates/**").addResourceLocations("classpath:/templates/"); super.addResourceHandlers(registry); } @Override public void addViewControllers(ViewControllerRegistry registry) { registry.addViewController("/index"); registry.addViewController("/user"); registry.addRedirectViewController("/","/templates/login.html"); // registry.addStatusController("/403", HttpStatus.FORBIDDEN); super.addViewControllers(registry); } }这里配置静态资源和直接访问的页面:在本示例项目中,添加了thymeleaf依赖解析视图,主要采用异步方式获取数据,通过angularJS进行前端数据的处理和展示。 使用activiti:配置了数据源和activiti后,启动项目,activiti 的各个服务组件就已经被加入到spring容器中了,所以就可以直接注入使用了。如果在未自动配置的spring环境中,可以使用通过指定bean的init-method来配置activiti的服务组件。 案例:以以下请假流程为例: 1. 开始流程并“申请请假”(员工) private static final String PROCESS_DEFINE_KEY = "vacationProcess"; public Object startVac(String userName, Vacation vac) { identityService.setAuthenticatedUserId(userName); // 开始流程 ProcessInstance vacationInstance = runtimeService.startProcessInstanceByKey(PROCESS_DEFINE_KEY); // 查询当前任务 Task currentTask = taskService.createTaskQuery().processInstanceId(vacationInstance.getId()).singleResult(); // 申明任务 taskService.claim(currentTask.getId(), userName); Map vars = new HashMap(4); vars.put("applyUser", userName); vars.put("days", vac.getDays()); vars.put("reason", vac.getReason()); // 完成任务 taskService.complete(currentTask.getId(), vars); return true; }在此方法中,Vaction 是申请时的具体信息,在完成“申请请假”任务时,可以将这些信息设置成参数。 2. 审批请假(老板) (1)查询需要自己审批的请假 public Object myAudit(String userName) { List taskList = taskService.createTaskQuery().taskCandidateUser(userName) .orderByTaskCreateTime().desc().list(); // / 多此一举 taskList中包含了以下内容(用户的任务中包含了所在用户组的任务) // Group group = identityService.createGroupQuery().groupMember(userName).singleResult(); // List list = taskService.createTaskQuery().taskCandidateGroup(group.getId()).list(); // taskList.addAll(list); List vacTaskList = new ArrayList(); for (Task task : taskList) { VacTask vacTask = new VacTask(); vacTask.setId(task.getId()); vacTask.setName(task.getName()); vacTask.setCreateTime(task.getCreateTime()); String instanceId = task.getProcessInstanceId(); ProcessInstance instance = runtimeService.createProcessInstanceQuery().processInstanceId(instanceId).singleResult(); Vacation vac = getVac(instance); vacTask.setVac(vac); vacTaskList.add(vacTask); } return vacTaskList; } private Vacation getVac(ProcessInstance instance) { Integer days = runtimeService.getVariable(instance.getId(), "days", Integer.class); String reason = runtimeService.getVariable(instance.getId(), "reason", String.class); Vacation vac = new Vacation(); vac.setApplyUser(instance.getStartUserId()); vac.setDays(days); vac.setReason(reason); Date startTime = instance.getStartTime(); // activiti 6 才有 vac.setApplyTime(startTime); vac.setApplyStatus(instance.isEnded() ? "申请结束" : "等待审批"); return vac; } package com.yawn.entity; import java.util.Date; /** * @author Created by yawn on 2018-01-09 14:31 */ public class VacTask { private String id; private String name; private Vacation vac; private Date createTime; // getter setter ... }老板查询自己当前需要审批的任务,并且将任务和参数设置到一个VacTask对象,用于页面的展示。 (2)审批请假 public Object passAudit(String userName, VacTask vacTask) { String taskId = vacTask.getId(); String result = vacTask.getVac().getResult(); Map vars = new HashMap(); vars.put("result", result); vars.put("auditor", userName); vars.put("auditTime", new Date()); taskService.claim(taskId, userName); taskService.complete(taskId, vars); return true; }同理,result是审批的结果,也是在完成审批任务时需要传入的参数;taskId是刚才老板查询到的当前需要自己完成的审批任务ID。(如果流程在这里设置分支,可以通过判断result的值来跳转到不同的任务) 3. 查询记录 由于已完成的请假在数据库runtime表中查不到(runtime表只保存正在进行的流程示例信息),所以需要在history表中查询。 (1) 查询请假记录 public Object myVacRecord(String userName) { List hisProInstance = historyService.createHistoricProcessInstanceQuery() .processDefinitionKey(PROCESS_DEFINE_KEY).startedBy(userName).finished() .orderByProcessInstanceEndTime().desc().list(); List vacList = new ArrayList(); for (HistoricProcessInstance hisInstance : hisProInstance) { Vacation vacation = new Vacation(); vacation.setApplyUser(hisInstance.getStartUserId()); vacation.setApplyTime(hisInstance.getStartTime()); vacation.setApplyStatus("申请结束"); List varInstanceList = historyService.createHistoricVariableInstanceQuery() .processInstanceId(hisInstance.getId()).list(); ActivitiUtil.setVars(vacation, varInstanceList); vacList.add(vacation); } return vacList; }请假记录即查出历史流程实例,再查出关联的历史参数,将历史流程实例和历史参数设置到Vcation对象(VO对象)中去,即可返回,用来展示。 package com.yawn.util; import org.activiti.engine.history.HistoricVariableInstance; import java.lang.reflect.Field; import java.util.List; /** * activiti中使用得到的工具方法 * @author Created by yawn on 2018-01-10 16:32 */ public class ActivitiUtil { /** * 将历史参数列表设置到实体中去 * @param entity 实体 * @param varInstanceList 历史参数列表 */ public static void setVars(T entity, List varInstanceList) { Class tClass = entity.getClass(); try { for (HistoricVariableInstance varInstance : varInstanceList) { Field field = tClass.getDeclaredField(varInstance.getVariableName()); if (field == null) { continue; } field.setAccessible(true); field.set(entity, varInstance.getValue()); } } catch (Exception e) { e.printStackTrace(); } } }此外,以上是查询历史流程实例和历史参数后,设置VO对象的通用方法:可以根据参数列表中的参数,将与VO对象属性同名的参数设置到VO对象中去。 4. 前端展示和操作 (1)审批列表和审批操作示例 待我审核的请假 任务名称 任务时间 申请人 申请时间 天数 事由 操作 {{vacTask.name}} {{vacTask.createTime | date:'yyyy-MM-dd HH:mm:ss'}} {{vacTask.vac.applyUser}} {{vacTask.vac.applyTime | date:'yyyy-MM-dd HH:mm:ss'}} {{vacTask.vac.days}} {{vacTask.vac.reason}} 审核通过 审核拒绝 app.controller("myAudit", function ($scope, $http, $window) { $scope.vacTaskList = []; $scope.myAudit = function () { $http.get( "/myAudit" ).then(function (response) { $scope.vacTaskList = response.data; }) }; $scope.passAudit = function (taskId, result) { $http.post( "/passAudit", { "id": taskId, "vac": { "result": result >= 1 ? "审核通过" : "审核拒绝" } } ).then(function (response) { if (response.data === true) { alert("操作成功!"); $window.location.reload(); } else { alert("操作失败!"); } }) } });以上是一个springBoot 与 activiti 6.0 整合的示例项目的部分代码与说明。 完整的项目代码在后台回复“整合”即可。 欢迎在留言区留下你的观点,一起讨论提高。如果今天的文章让你有新的启发,学习能力的提升上有新的认识,欢迎转发分享给更多人。 欢迎各位读者加入程序员小乐技术群,在公众号后台回复“加群”或者“学习”即可。 猜你还想看 阿里、腾讯、百度、华为、京东最新面试题汇集 图解排序算法:彻底搞懂归并排序 Java 配合 mitmproxy HTTPS 抓包调试 93.7% 的程序员,竟然都不知道 Redis 为什么默认16个数据库? 关注「程序员小乐」,收看更多精彩内容嘿,你在看吗? |
CopyRight 2018-2019 实验室设备网 版权所有 |